/*
Copyright 1999 CERN - European Organization for Nuclear Research.
Permission to use, copy, modify, distribute and sell this software and its documentation for any purpose 
is hereby granted without fee, provided that the above copyright notice appear in all copies and 
that both that copyright notice and this permission notice appear in supporting documentation. 
CERN makes no representations about the suitability of this software for any purpose. 
It is provided "as is" without expressed or implied warranty.
*/
package com.zongtui.classifier.math;

/**
 * Array manipulations; complements <tt>java.util.Arrays</tt>.
 *
 * @see java.util.Arrays
 * @see org.apache.mahout.math.Sorting
 *
 */
public final class Arrays {

  private Arrays() {
  }

  /**
   * Ensures that a given array can hold up to <tt>minCapacity</tt> elements.
   *
   * Returns the identical array if it can hold at least the number of elements specified. Otherwise, returns a new
   * array with increased capacity containing the same elements, ensuring that it can hold at least the number of
   * elements specified by the minimum capacity argument.
   *
   * @param minCapacity the desired minimum capacity.
   */
  public static byte[] ensureCapacity(byte[] array, int minCapacity) {
    int oldCapacity = array.length;
    byte[] newArray;
    if (minCapacity > oldCapacity) {
      int newCapacity = (oldCapacity * 3) / 2 + 1;
      if (newCapacity < minCapacity) {
        newCapacity = minCapacity;
      }

      newArray = new byte[newCapacity];
      System.arraycopy(array, 0, newArray, 0, oldCapacity);
    } else {
      newArray = array;
    }
    return newArray;
  }

  /**
   * Ensures that a given array can hold up to <tt>minCapacity</tt> elements.
   *
   * Returns the identical array if it can hold at least the number of elements specified. Otherwise, returns a new
   * array with increased capacity containing the same elements, ensuring that it can hold at least the number of
   * elements specified by the minimum capacity argument.
   *
   * @param minCapacity the desired minimum capacity.
   */
  public static char[] ensureCapacity(char[] array, int minCapacity) {
    int oldCapacity = array.length;
    char[] newArray;
    if (minCapacity > oldCapacity) {
      int newCapacity = (oldCapacity * 3) / 2 + 1;
      if (newCapacity < minCapacity) {
        newCapacity = minCapacity;
      }

      newArray = new char[newCapacity];
      System.arraycopy(array, 0, newArray, 0, oldCapacity);
    } else {
      newArray = array;
    }
    return newArray;
  }

  /**
   * Ensures that a given array can hold up to <tt>minCapacity</tt> elements.
   *
   * Returns the identical array if it can hold at least the number of elements specified. Otherwise, returns a new
   * array with increased capacity containing the same elements, ensuring that it can hold at least the number of
   * elements specified by the minimum capacity argument.
   *
   * @param minCapacity the desired minimum capacity.
   */
  public static double[] ensureCapacity(double[] array, int minCapacity) {
    int oldCapacity = array.length;
    double[] newArray;
    if (minCapacity > oldCapacity) {
      int newCapacity = (oldCapacity * 3) / 2 + 1;
      if (newCapacity < minCapacity) {
        newCapacity = minCapacity;
      }

      newArray = new double[newCapacity];
      //for (int i = oldCapacity; --i >= 0; ) newArray[i] = array[i];
      System.arraycopy(array, 0, newArray, 0, oldCapacity);
    } else {
      newArray = array;
    }
    return newArray;
  }

  /**
   * Ensures that a given array can hold up to <tt>minCapacity</tt> elements.
   *
   * Returns the identical array if it can hold at least the number of elements specified. Otherwise, returns a new
   * array with increased capacity containing the same elements, ensuring that it can hold at least the number of
   * elements specified by the minimum capacity argument.
   *
   * @param minCapacity the desired minimum capacity.
   */
  public static float[] ensureCapacity(float[] array, int minCapacity) {
    int oldCapacity = array.length;
    float[] newArray;
    if (minCapacity > oldCapacity) {
      int newCapacity = (oldCapacity * 3) / 2 + 1;
      if (newCapacity < minCapacity) {
        newCapacity = minCapacity;
      }

      newArray = new float[newCapacity];
      System.arraycopy(array, 0, newArray, 0, oldCapacity);
    } else {
      newArray = array;
    }
    return newArray;
  }

  /**
   * Ensures that a given array can hold up to <tt>minCapacity</tt> elements.
   *
   * Returns the identical array if it can hold at least the number of elements specified. Otherwise, returns a new
   * array with increased capacity containing the same elements, ensuring that it can hold at least the number of
   * elements specified by the minimum capacity argument.
   *
   * @param minCapacity the desired minimum capacity.
   */
  public static int[] ensureCapacity(int[] array, int minCapacity) {
    int oldCapacity = array.length;
    int[] newArray;
    if (minCapacity > oldCapacity) {
      int newCapacity = (oldCapacity * 3) / 2 + 1;
      if (newCapacity < minCapacity) {
        newCapacity = minCapacity;
      }

      newArray = new int[newCapacity];
      System.arraycopy(array, 0, newArray, 0, oldCapacity);
    } else {
      newArray = array;
    }
    return newArray;
  }

  /**
   * Ensures that a given array can hold up to <tt>minCapacity</tt> elements.
   *
   * Returns the identical array if it can hold at least the number of elements specified. Otherwise, returns a new
   * array with increased capacity containing the same elements, ensuring that it can hold at least the number of
   * elements specified by the minimum capacity argument.
   *
   * @param minCapacity the desired minimum capacity.
   */
  public static long[] ensureCapacity(long[] array, int minCapacity) {
    int oldCapacity = array.length;
    long[] newArray;
    if (minCapacity > oldCapacity) {
      int newCapacity = (oldCapacity * 3) / 2 + 1;
      if (newCapacity < minCapacity) {
        newCapacity = minCapacity;
      }

      newArray = new long[newCapacity];
      System.arraycopy(array, 0, newArray, 0, oldCapacity);
    } else {
      newArray = array;
    }
    return newArray;
  }

  /**
   * Ensures that a given array can hold up to <tt>minCapacity</tt> elements.
   *
   * Returns the identical array if it can hold at least the number of elements specified. Otherwise, returns a new
   * array with increased capacity containing the same elements, ensuring that it can hold at least the number of
   * elements specified by the minimum capacity argument.
   *
   * @param minCapacity the desired minimum capacity.
   */
  public static Object[] ensureCapacity(Object[] array, int minCapacity) {
    int oldCapacity = array.length;
    Object[] newArray;
    if (minCapacity > oldCapacity) {
      int newCapacity = (oldCapacity * 3) / 2 + 1;
      if (newCapacity < minCapacity) {
        newCapacity = minCapacity;
      }

      newArray = new Object[newCapacity];
      System.arraycopy(array, 0, newArray, 0, oldCapacity);
    } else {
      newArray = array;
    }
    return newArray;
  }

  /**
   * Ensures that a given array can hold up to <tt>minCapacity</tt> elements.
   *
   * Returns the identical array if it can hold at least the number of elements specified. Otherwise, returns a new
   * array with increased capacity containing the same elements, ensuring that it can hold at least the number of
   * elements specified by the minimum capacity argument.
   *
   * @param minCapacity the desired minimum capacity.
   */
  public static short[] ensureCapacity(short[] array, int minCapacity) {
    int oldCapacity = array.length;
    short[] newArray;
    if (minCapacity > oldCapacity) {
      int newCapacity = (oldCapacity * 3) / 2 + 1;
      if (newCapacity < minCapacity) {
        newCapacity = minCapacity;
      }

      newArray = new short[newCapacity];
      System.arraycopy(array, 0, newArray, 0, oldCapacity);
    } else {
      newArray = array;
    }
    return newArray;
  }

  /**
   * Ensures that a given array can hold up to <tt>minCapacity</tt> elements.
   *
   * Returns the identical array if it can hold at least the number of elements specified. Otherwise, returns a new
   * array with increased capacity containing the same elements, ensuring that it can hold at least the number of
   * elements specified by the minimum capacity argument.
   *
   * @param minCapacity the desired minimum capacity.
   */
  public static boolean[] ensureCapacity(boolean[] array, int minCapacity) {
    int oldCapacity = array.length;
    boolean[] newArray;
    if (minCapacity > oldCapacity) {
      int newCapacity = (oldCapacity * 3) / 2 + 1;
      if (newCapacity < minCapacity) {
        newCapacity = minCapacity;
      }

      newArray = new boolean[newCapacity];
      System.arraycopy(array, 0, newArray, 0, oldCapacity);
    } else {
      newArray = array;
    }
    return newArray;
  }

  /**
   * Returns a string representation of the specified array.  The string representation consists of a list of the
   * arrays's elements, enclosed in square brackets (<tt>"[]"</tt>).  Adjacent elements are separated by the characters
   * <tt>", "</tt> (comma and space).
   *
   * @return a string representation of the specified array.
   */
  public static String toString(byte[] array) {
    StringBuilder buf = new StringBuilder();
    buf.append('[');
    int maxIndex = array.length - 1;
    for (int i = 0; i <= maxIndex; i++) {
      buf.append(array[i]);
      if (i < maxIndex) {
        buf.append(", ");
      }
    }
    buf.append(']');
    return buf.toString();
  }

  /**
   * Returns a string representation of the specified array.  The string representation consists of a list of the
   * arrays's elements, enclosed in square brackets (<tt>"[]"</tt>).  Adjacent elements are separated by the characters
   * <tt>", "</tt> (comma and space).
   *
   * @return a string representation of the specified array.
   */
  public static String toString(char[] array) {
    StringBuilder buf = new StringBuilder();
    buf.append('[');
    int maxIndex = array.length - 1;
    for (int i = 0; i <= maxIndex; i++) {
      buf.append(array[i]);
      if (i < maxIndex) {
        buf.append(", ");
      }
    }
    buf.append(']');
    return buf.toString();
  }

  /**
   * Returns a string representation of the specified array.  The string representation consists of a list of the
   * arrays's elements, enclosed in square brackets (<tt>"[]"</tt>).  Adjacent elements are separated by the characters
   * <tt>", "</tt> (comma and space).
   *
   * @return a string representation of the specified array.
   */
  public static String toString(double[] array) {
    StringBuilder buf = new StringBuilder();
    buf.append('[');
    int maxIndex = array.length - 1;
    for (int i = 0; i <= maxIndex; i++) {
      buf.append(array[i]);
      if (i < maxIndex) {
        buf.append(", ");
      }
    }
    buf.append(']');
    return buf.toString();
  }

  /**
   * Returns a string representation of the specified array.  The string representation consists of a list of the
   * arrays's elements, enclosed in square brackets (<tt>"[]"</tt>).  Adjacent elements are separated by the characters
   * <tt>", "</tt> (comma and space).
   *
   * @return a string representation of the specified array.
   */
  public static String toString(float[] array) {
    StringBuilder buf = new StringBuilder();
    buf.append('[');
    int maxIndex = array.length - 1;
    for (int i = 0; i <= maxIndex; i++) {
      buf.append(array[i]);
      if (i < maxIndex) {
        buf.append(", ");
      }
    }
    buf.append(']');
    return buf.toString();
  }

  /**
   * Returns a string representation of the specified array.  The string representation consists of a list of the
   * arrays's elements, enclosed in square brackets (<tt>"[]"</tt>).  Adjacent elements are separated by the characters
   * <tt>", "</tt> (comma and space).
   *
   * @return a string representation of the specified array.
   */
  public static String toString(int[] array) {
    StringBuilder buf = new StringBuilder();
    buf.append('[');
    int maxIndex = array.length - 1;
    for (int i = 0; i <= maxIndex; i++) {
      buf.append(array[i]);
      if (i < maxIndex) {
        buf.append(", ");
      }
    }
    buf.append(']');
    return buf.toString();
  }

  /**
   * Returns a string representation of the specified array.  The string representation consists of a list of the
   * arrays's elements, enclosed in square brackets (<tt>"[]"</tt>).  Adjacent elements are separated by the characters
   * <tt>", "</tt> (comma and space).
   *
   * @return a string representation of the specified array.
   */
  public static String toString(long[] array) {
    StringBuilder buf = new StringBuilder();
    buf.append('[');
    int maxIndex = array.length - 1;
    for (int i = 0; i <= maxIndex; i++) {
      buf.append(array[i]);
      if (i < maxIndex) {
        buf.append(", ");
      }
    }
    buf.append(']');
    return buf.toString();
  }

  /**
   * Returns a string representation of the specified array.  The string representation consists of a list of the
   * arrays's elements, enclosed in square brackets (<tt>"[]"</tt>).  Adjacent elements are separated by the characters
   * <tt>", "</tt> (comma and space).
   *
   * @return a string representation of the specified array.
   */
  public static String toString(Object[] array) {
    StringBuilder buf = new StringBuilder();
    buf.append('[');
    int maxIndex = array.length - 1;
    for (int i = 0; i <= maxIndex; i++) {
      buf.append(array[i]);
      if (i < maxIndex) {
        buf.append(", ");
      }
    }
    buf.append(']');
    return buf.toString();
  }

  /**
   * Returns a string representation of the specified array.  The string representation consists of a list of the
   * arrays's elements, enclosed in square brackets (<tt>"[]"</tt>).  Adjacent elements are separated by the characters
   * <tt>", "</tt> (comma and space).
   *
   * @return a string representation of the specified array.
   */
  public static String toString(short[] array) {
    StringBuilder buf = new StringBuilder();
    buf.append('[');
    int maxIndex = array.length - 1;
    for (int i = 0; i <= maxIndex; i++) {
      buf.append(array[i]);
      if (i < maxIndex) {
        buf.append(", ");
      }
    }
    buf.append(']');
    return buf.toString();
  }

  /**
   * Returns a string representation of the specified array.  The string representation consists of a list of the
   * arrays's elements, enclosed in square brackets (<tt>"[]"</tt>).  Adjacent elements are separated by the characters
   * <tt>", "</tt> (comma and space).
   *
   * @return a string representation of the specified array.
   */
  public static String toString(boolean[] array) {
    StringBuilder buf = new StringBuilder();
    buf.append('[');
    int maxIndex = array.length - 1;
    for (int i = 0; i <= maxIndex; i++) {
      buf.append(array[i]);
      if (i < maxIndex) {
        buf.append(", ");
      }
    }
    buf.append(']');
    return buf.toString();
  }

  /**
   * Ensures that the specified array cannot hold more than <tt>maxCapacity</tt> elements. An application can use this
   * operation to minimize array storage. <p> Returns the identical array if <tt>array.length &lt;= maxCapacity</tt>.
   * Otherwise, returns a new array with a length of <tt>maxCapacity</tt> containing the first <tt>maxCapacity</tt>
   * elements of <tt>array</tt>.
   *
   * @param maxCapacity the desired maximum capacity.
   */
  public static byte[] trimToCapacity(byte[] array, int maxCapacity) {
    if (array.length > maxCapacity) {
      byte[] oldArray = array;
      array = new byte[maxCapacity];
      System.arraycopy(oldArray, 0, array, 0, maxCapacity);
    }
    return array;
  }

  /**
   * Ensures that the specified array cannot hold more than <tt>maxCapacity</tt> elements. An application can use this
   * operation to minimize array storage. <p> Returns the identical array if <tt>array.length &lt;= maxCapacity</tt>.
   * Otherwise, returns a new array with a length of <tt>maxCapacity</tt> containing the first <tt>maxCapacity</tt>
   * elements of <tt>array</tt>.
   *
   * @param maxCapacity the desired maximum capacity.
   */
  public static char[] trimToCapacity(char[] array, int maxCapacity) {
    if (array.length > maxCapacity) {
      char[] oldArray = array;
      array = new char[maxCapacity];
      System.arraycopy(oldArray, 0, array, 0, maxCapacity);
    }
    return array;
  }

  /**
   * Ensures that the specified array cannot hold more than <tt>maxCapacity</tt> elements. An application can use this
   * operation to minimize array storage. <p> Returns the identical array if <tt>array.length &lt;= maxCapacity</tt>.
   * Otherwise, returns a new array with a length of <tt>maxCapacity</tt> containing the first <tt>maxCapacity</tt>
   * elements of <tt>array</tt>.
   *
   * @param maxCapacity the desired maximum capacity.
   */
  public static double[] trimToCapacity(double[] array, int maxCapacity) {
    if (array.length > maxCapacity) {
      double[] oldArray = array;
      array = new double[maxCapacity];
      System.arraycopy(oldArray, 0, array, 0, maxCapacity);
    }
    return array;
  }

  /**
   * Ensures that the specified array cannot hold more than <tt>maxCapacity</tt> elements. An application can use this
   * operation to minimize array storage. <p> Returns the identical array if <tt>array.length &lt;= maxCapacity</tt>.
   * Otherwise, returns a new array with a length of <tt>maxCapacity</tt> containing the first <tt>maxCapacity</tt>
   * elements of <tt>array</tt>.
   *
   * @param maxCapacity the desired maximum capacity.
   */
  public static float[] trimToCapacity(float[] array, int maxCapacity) {
    if (array.length > maxCapacity) {
      float[] oldArray = array;
      array = new float[maxCapacity];
      System.arraycopy(oldArray, 0, array, 0, maxCapacity);
    }
    return array;
  }

  /**
   * Ensures that the specified array cannot hold more than <tt>maxCapacity</tt> elements. An application can use this
   * operation to minimize array storage. <p> Returns the identical array if <tt>array.length &lt;= maxCapacity</tt>.
   * Otherwise, returns a new array with a length of <tt>maxCapacity</tt> containing the first <tt>maxCapacity</tt>
   * elements of <tt>array</tt>.
   *
   * @param maxCapacity the desired maximum capacity.
   */
  public static int[] trimToCapacity(int[] array, int maxCapacity) {
    if (array.length > maxCapacity) {
      int[] oldArray = array;
      array = new int[maxCapacity];
      System.arraycopy(oldArray, 0, array, 0, maxCapacity);
    }
    return array;
  }

  /**
   * Ensures that the specified array cannot hold more than <tt>maxCapacity</tt> elements. An application can use this
   * operation to minimize array storage. <p> Returns the identical array if <tt>array.length &lt;= maxCapacity</tt>.
   * Otherwise, returns a new array with a length of <tt>maxCapacity</tt> containing the first <tt>maxCapacity</tt>
   * elements of <tt>array</tt>.
   *
   * @param maxCapacity the desired maximum capacity.
   */
  public static long[] trimToCapacity(long[] array, int maxCapacity) {
    if (array.length > maxCapacity) {
      long[] oldArray = array;
      array = new long[maxCapacity];
      System.arraycopy(oldArray, 0, array, 0, maxCapacity);
    }
    return array;
  }

  /**
   * Ensures that the specified array cannot hold more than <tt>maxCapacity</tt> elements. An application can use this
   * operation to minimize array storage. <p> Returns the identical array if <tt>array.length &lt;= maxCapacity</tt>.
   * Otherwise, returns a new array with a length of <tt>maxCapacity</tt> containing the first <tt>maxCapacity</tt>
   * elements of <tt>array</tt>.
   *
   * @param maxCapacity the desired maximum capacity.
   */
  public static Object[] trimToCapacity(Object[] array, int maxCapacity) {
    if (array.length > maxCapacity) {
      Object[] oldArray = array;
      array = new Object[maxCapacity];
      System.arraycopy(oldArray, 0, array, 0, maxCapacity);
    }
    return array;
  }

  /**
   * Ensures that the specified array cannot hold more than <tt>maxCapacity</tt> elements. An application can use this
   * operation to minimize array storage. <p> Returns the identical array if <tt>array.length &lt;= maxCapacity</tt>.
   * Otherwise, returns a new array with a length of <tt>maxCapacity</tt> containing the first <tt>maxCapacity</tt>
   * elements of <tt>array</tt>.
   *
   * @param maxCapacity the desired maximum capacity.
   */
  public static short[] trimToCapacity(short[] array, int maxCapacity) {
    if (array.length > maxCapacity) {
      short[] oldArray = array;
      array = new short[maxCapacity];
      System.arraycopy(oldArray, 0, array, 0, maxCapacity);
    }
    return array;
  }

  /**
   * Ensures that the specified array cannot hold more than <tt>maxCapacity</tt> elements. An application can use this
   * operation to minimize array storage. <p> Returns the identical array if <tt>array.length &lt;= maxCapacity</tt>.
   * Otherwise, returns a new array with a length of <tt>maxCapacity</tt> containing the first <tt>maxCapacity</tt>
   * elements of <tt>array</tt>.
   *
   * @param maxCapacity the desired maximum capacity.
   */
  public static boolean[] trimToCapacity(boolean[] array, int maxCapacity) {
    if (array.length > maxCapacity) {
      boolean[] oldArray = array;
      array = new boolean[maxCapacity];
      System.arraycopy(oldArray, 0, array, 0, maxCapacity);
    }
    return array;
  }

  /**
   * {@link java.util.Arrays#copyOf} compatibility with Java 1.5.
   */
  public static byte[] copyOf(byte[] src, int length) {
    byte[] result = new byte [length];
    System.arraycopy(src, 0, result, 0, Math.min(length, src.length));
    return result;
  }
  
  /**
   * {@link java.util.Arrays#copyOf} compatibility with Java 1.5.
   */
  public static char[] copyOf(char[] src, int length) {
    char[] result = new char [length];
    System.arraycopy(src, 0, result, 0, Math.min(length, src.length));
    return result;
  }
  
  /**
   * {@link java.util.Arrays#copyOf} compatibility with Java 1.5.
   */
  public static short[] copyOf(short[] src, int length) {
    short[] result = new short [length];
    System.arraycopy(src, 0, result, 0, Math.min(length, src.length));
    return result;
  }
  
  /**
   * {@link java.util.Arrays#copyOf} compatibility with Java 1.5. 
   */
  public static int[] copyOf(int[] src, int length) {
    int[] result = new int [length];
    System.arraycopy(src, 0, result, 0, Math.min(length, src.length));
    return result;
  }
  
  /**
   * {@link java.util.Arrays#copyOf} compatibility with Java 1.5.
   */
  public static float[] copyOf(float[] src, int length) {
    float[] result = new float [length];
    System.arraycopy(src, 0, result, 0, Math.min(length, src.length));
    return result;
  }

  /**
   * {@link java.util.Arrays#copyOf} compatibility with Java 1.5.
   */
  public static double[] copyOf(double[] src, int length) {
    double[] result = new double [length];
    System.arraycopy(src, 0, result, 0, Math.min(length, src.length));
    return result;
  }
  
  /**
   * {@link java.util.Arrays#copyOf} compatibility with Java 1.5.
   */
  public static long[] copyOf(long[] src, int length) {
    long[] result = new long [length];
    System.arraycopy(src, 0, result, 0, Math.min(length, src.length));
    return result;
  }
}
